Android 6.0 运行时权限常识

如果APP运行在Android 6.0或以上版本的手机,并且target sdk>=23,那么在使用一些相对敏感的权限时,需要征求用户的许可。比如读写sdcard,摄像,联系人信息等。我们在开发时,如果指定了系统的最高运行版本低于23( 比如targetSdkVersion 21),就算是运行在6.0手机也不会出现权限提示框的。

request权限 如图所示:
request权限
android 6.0发布几个月之后,升级率很低,很少应用会使用sdk23编译项目,如果要编译,就必须了解request权限的一些事。

一,Android 6.0 运行时权限

Android 6.0在原有的AndroidManifest.xml声明权限的基础上,新增了运行时权限动态检测,以下权限都需要在运行时判断:

  1. 身体传感器
  2. 日历 摄像头
  3. 通讯录
  4. 地理位置
  5. 麦克风
  6. 电话
  7. 短信
  8. 存储空间
  9. 运行时权限处理
    PS:targetSdkVersion小于23的应用默认授予了所申请的所有权限,所以如果你以前的APP设置的targetSdkVersion低于23,也能正常使用。等于或者大于23,则必须 request permission。

    二,声明目标SDK版本

    我们需要在build.gradle中声明targetSdkVersion为23.
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    android {
    compileSdkVersion 23
    buildToolsVersion "23.0.2"
    defaultConfig {
    applicationId "com.youapp"
    minSdkVersion 14
    targetSdkVersion 23
    versionCode 1
    versionName "1.0"
    }

三,检查并request权限

我们需要在用到权限的地方,检查是否已经拥有权限,比如读写外置SD卡的权限,我们在写入之前检查是否有权限,没有则申请权限,for exsample:

1
2
3
4
5
6
7
8
private void requestContactPermission() {
if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_CONTACTS)
!= PackageManager.PERMISSION_GRANTED) {
//申请 WRITE_CONTACTS 权限
ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_CONTACTS},
REQUEST_CODE_WRITE_CONTACTS);
}
}

四,用户许可与否的回调处理。

权限申请后,用户选择后,可能allow,可能deny。回调onRequestPermissionsResult方法, 该方法类似于onActivityResult。如果是fragment,最好是使用父fragment,但不是使用ActivityCompat。建议使用getParentFragment().requestPermissions方法。
兼容库support-v4中的方法有:
ContextCompat.checkSelfPermission()
ActivityCompat.requestPermissions()
ActivityCompat.OnRequestPermissionsResultCallback
ActivityCompat.shouldShowRequestPermissionRationale()

1
2
3
4
5
6
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
handleGrantResults(requestCode,grantResults);
}

五,根据requestCode和grantResults(授权结果)做相应的处理。

1
2
3
4
5
6
7
8
9
private void handleGrantResults(int requestCode, int[] grantResults) {
if (requestCode == WRITE_EXTERNAL_STORAGE_REQUEST_CODE) {
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
// Permission Granted 获得权限后执行xxx
} else {
// Permission Denied 拒绝后xx的操作。
}
}
}

六,用户拒绝了怎么处理?

如果用户拒绝了一次,后面再申请,会出现不再提醒的选项,再次拒绝,你的app不作后续处理,就嗝屁了。功能肯定受限,没法用了。
拒绝后在此提示,有never ask again
这时候怎么办呢?用户的行为我们不清楚,得预防一下。
我们检测下是否拒绝了两次或以上,DIY给个提示。比如申请写联系人的权限。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
private void handleContactPermission() {
if (Integer.parseInt(Build.VERSION.SDK)>=23) {
int hasWriteContactsPermission = checkSelfPermission(Manifest.permission.WRITE_CONTACTS);
if (hasWriteContactsPermission != PackageManager.PERMISSION_GRANTED) {
if (!shouldShowRequestPermissionRationale(Manifest.permission.WRITE_CONTACTS)) {
showMessageOKCancel("You need to allow access to Contacts",
new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
requestContactPermission();//确定后申请权限。
}
});
return;
}
requestContactPermission();//没有权限的话,申请。
}
}

效果如下:
提示用户必须允许访问

七,权限组permission-group

dangerous permissions 被分入到 9 个权限组.
同一组的任何一个权限被授权了,其他权限也自动被授权。例如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。
permission-group

github已经有一些封装好的permissionUtil,大家可以下载学习。不足之处,后续完善。
Dusan,291902259,杜工。